-
Notifications
You must be signed in to change notification settings - Fork 726
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add file storage option #12590
base: develop
Are you sure you want to change the base?
Add file storage option #12590
Conversation
Build Artifacts
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think we probably want to keep the options.py interface restricted, others can override settings if they see fit, but we should focus on enabling the specific options we need.
Hint: if you do kolibri configure list-env
you will see a complete list of available env vars for configuration (which will also show how your new option can be set as an env var).
kolibri/utils/options.py
Outdated
@@ -359,6 +377,16 @@ def lazy_import_callback_list(value): | |||
|
|||
|
|||
base_option_spec = { | |||
"FileStorage": { | |||
"DEFAULT_FILE_STORAGE": { |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
I think I'd want to hew closer to the pattern we have for the Cache and Database options here - and offer simple, pre-specified string options that refer to specific backends. If someone really wants to run a custom backend, they can override the settings file and do what they like.
That way, with specific backends in mind, we can then explicitly enumerate the additional things that need to be specified in each case - for example the default "file_system"
backend value will need a path, if it's a GCS backend then other things might be required (or may be automagically configured in some cases).
kolibri/utils/options.py
Outdated
except ImportError: | ||
logger.error("Default file storage is not available.") | ||
raise VdtValueError(value) | ||
except Exception: |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Shouldn't ever be catching a bare Exception, unless for very good reason - it can hide a multitude of sins.
kolibri/utils/options.py
Outdated
modules = value.split(".") | ||
klass = modules.pop() | ||
module_path = ".".join(modules) | ||
module = importlib.import_module(module_path) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Note that Django exposes a utility called import_string
which we already use in this module for loading classes by a string dot path, so this seems preferable to use here.
kolibri/utils/options.py
Outdated
@@ -15,6 +15,7 @@ | |||
from configobj import ConfigObj | |||
from configobj import flatten_errors | |||
from configobj import get_extra_values | |||
from django.core.files.storage import Storage |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
This is probably why flake8 wouldn't let pre-commit pass but it didn't give me useful output
@@ -737,6 +766,7 @@ def _get_validator(): | |||
"url_prefix": url_prefix, | |||
"bytes": validate_bytes, | |||
"multiprocess_bool": multiprocess_bool, | |||
"storage_option": storage_option, |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
how will this work with database based cache?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Hm - I'm not sure. I assumed this was only related to validation on initialization
logger.info("File saved - Path: {}".format(file_storage.url(file))) | ||
logger.info("File saved - Size: {}".format(file_storage.size(file))) |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
@jredrejo I've tried several things around here, but no matter what I do I the file always shows size 0... I've confirmed that there are users (usernames is full of data, for example) -- so I'm not sure why the writer
isn't updating the file object here...
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The problem was not in reading the file info, but writing it. It had 0 bytes. I have made a commit in this PR to ensure data is flushed into the BytesIO and now it's working:
INFO 2024-12-26 20:45:45,313 Invoking command bulkexportusers
INFO 2024-12-26 20:45:45,373 Creating users csv file /home/jose/.kolibri/log_export/Kolibri en casa de admin_3da4_users.csv
[#######################################################################################################################################################################################-------------------------------------------------------------] 75%INFO 2024-12-26 20:45:46,767 File saved - Path: https://storage.googleapis.com/kdp-csv-reporting-develop/Kolibri_en_casa_de_admin_3da4_users.csv
INFO 2024-12-26 20:45:47,073 File saved - Size: 482
INFO 2024-12-26 20:45:47,073 Created csv file /home/jose/.kolibri/log_export/Kolibri en casa de admin_3da4_users.csv with 4 lines
(I've also rebased develop in the branch because there was already a conflict and more might appear soon)
…FAULT_FILE_STORAGE
…underlying BytesIO buffer
43309bc
to
953b29b
Compare
Summary
In order to allow us to use a cloud backend for the File Storage in Django, this adds a value to options.py which defaults to the Django FileSystemStorage (which... it did anyway but now we make sure of it).
This should make it configurable on BCK such that if there is a module that is a Google cloud backend class that implements the Django Storage class, then it can be added as the value for the settings.
For example, if we have a new class "GCloudStorage" in
kolibri.core.storage
then we would use that class if we set the option added here tokolibri.core.storage.GCloudStorage
.This is very much a first whack -- one thing I'm not clear on is if by naming the option by the name that Django would look to in the env vars
DEFAULT_FILE_STORAGE
does the Kolibri options.py stuff automatically apply that setting because of the matching name?References
Fixes #9441 (or at least begins to address it)
Reviewer guidance
STORAGE_BACKEND
and change it's value togcs
and restart Kolibri, you should get an error message that the module is not availablepip install -r requirements/storages.txt
then try again, no error and it should start upNext steps
Testing checklist
PR process
Reviewer checklist
yarn
andpip
)